package com.tongcao.cn.util;


import com.xiaoleilu.hutool.io.FileUtil;
import com.xiaoleilu.hutool.io.IORuntimeException;
import com.xiaoleilu.hutool.util.StrUtil;

import javax.imageio.ImageIO;
import javax.imageio.ImageReader;
import javax.imageio.stream.ImageOutputStream;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Iterator;

/**
 * 图片处理工具类：<br>
 * 功能：缩放图像、切割图像、旋转、图像类型转换、彩色转黑白、文字水印、图片水印等 <br>
 * 参考：http://blog.csdn.net/zhangzhikaixinya/article/details/8459400
 *
 * @author Looly
 */
public class ImageUtil {

    public static final String IMAGE_TYPE_GIF = "gif";// 图形交换格式
    public static final String IMAGE_TYPE_JPG = "jpg";// 联合照片专家组
    public static final String IMAGE_TYPE_JPEG = "jpeg";// 联合照片专家组
    public static final String IMAGE_TYPE_BMP = "bmp";// 英文Bitmap（位图）的简写，它是Windows操作系统中的标准图像文件格式
    public static final String IMAGE_TYPE_PNG = "png";// 可移植网络图形
    public static final String IMAGE_TYPE_PSD = "psd";// Photoshop的专用格式Photoshop

    // ---------------------------------------------------------------------------------------------------------------------- scale
//	/**
//	 * 缩放图像（按比例缩放）<br>
//	 * 缩放后默认为jpeg格式
//	 *
//	 * @param srcImageFile 源图像文件
//	 * @param destImageFile 缩放后的图像文件
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 */
//	public static void scale(File srcImageFile, File destImageFile, float scale) {
//		scale(read(srcImageFile), destImageFile, scale);
//	}
//
//	/**
//	 * 缩放图像（按比例缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcStream 源图像来源流
//	 * @param destStream 缩放后的图像写出到的流
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 * @since 3.0.9
//	 */
//	public static void scale(InputStream srcStream, OutputStream destStream, float scale) {
//		scale(read(srcStream), destStream, scale);
//	}
//
//	/**
//	 * 缩放图像（按比例缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcStream 源图像来源流
//	 * @param destStream 缩放后的图像写出到的流
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 * @since 3.1.0
//	 */
//	public static void scale(ImageInputStream srcStream, ImageOutputStream destStream, float scale) {
//		scale(read(srcStream), destStream, scale);
//	}

//	/**
//	 * 缩放图像（按比例缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcImg 源图像来源流
//	 * @param destFile 缩放后的图像写出到的流
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void scale(Image srcImg, File destFile, float scale) throws IORuntimeException {
//		write(scale(srcImg, scale), destFile);
//	}

//	/**
//	 * 缩放图像（按比例缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcImg 源图像来源流
//	 * @param out 缩放后的图像写出到的流
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void scale(Image srcImg, OutputStream out, float scale) throws IORuntimeException {
//		scale(srcImg, getImageOutputStream(out), scale);
//	}
//
//	/**
//	 * 缩放图像（按比例缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcImg 源图像来源流
//	 * @param destImageStream 缩放后的图像写出到的流
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 * @throws IORuntimeException IO异常
//	 * @since 3.1.0
//	 */
//	public static void scale(Image srcImg, ImageOutputStream destImageStream, float scale) throws IORuntimeException {
//		writeJpg(scale(srcImg, scale), destImageStream);
//	}
//
//	/**
//	 * 缩放图像（按比例缩放）
//	 *
//	 * @param srcImg 源图像来源流
//	 * @param scale 缩放比例。比例大于1时为放大，小于1大于0为缩小
//	 * @return {@link Image}
//	 * @since 3.1.0
//	 */
//	public static Image scale(Image srcImg, float scale) {
//		if (scale < 0) {
//			// 自动修正负数
//			scale = -scale;
//		}
//
//		int width = NumberUtil.mul(Integer.toString(srcImg.getWidth(null)), Float.toString(scale)).intValue(); // 得到源图宽
//		int height = NumberUtil.mul(Integer.toString(srcImg.getHeight(null)), Float.toString(scale)).intValue(); // 得到源图长
//		return scale(srcImg, width, height);
//	}
//
//	/**
//	 * 缩放图像（按长宽缩放）<br>
//	 * 注意：目标长宽与原图不成比例会变形
//	 *
//	 * @param srcImg 源图像来源流
//	 * @param width 目标宽度
//	 * @param height 目标高度
//	 * @return {@link Image}
//	 * @since 3.1.0
//	 */
//	public static Image scale(Image srcImg, int width, int height) {
//		int srcHeight = srcImg.getHeight(null);
//		int srcWidth = srcImg.getWidth(null);
//		int scaleType;
//		if (srcHeight == height && srcWidth == width) {
//			// 源与目标长宽一致返回原图
//			return srcImg;
//		} else if (srcHeight < height || srcWidth < width) {
//			// 放大图片使用平滑模式
//			scaleType = Image.SCALE_SMOOTH;
//		} else {
//			scaleType = Image.SCALE_DEFAULT;
//		}
//		return srcImg.getScaledInstance(width, height, scaleType);
//	}
//
//	/**
//	 * 缩放图像（按高度和宽度缩放）<br>
//	 * 缩放后默认为jpeg格式
//	 *
//	 * @param srcImageFile 源图像文件地址
//	 * @param destImageFile 缩放后的图像地址
//	 * @param width 缩放后的宽度
//	 * @param height 缩放后的高度
//	 * @param fixedColor 比例不对时补充的颜色，不补充为<code>null</code>
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void scale(File srcImageFile, File destImageFile, int width, int height, Color fixedColor) throws IORuntimeException {
//		write(scale(read(srcImageFile), width, height, fixedColor), destImageFile);
//	}
//
//	/**
//	 * 缩放图像（按高度和宽度缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 缩放后的图像目标流
//	 * @param width 缩放后的宽度
//	 * @param height 缩放后的高度
//	 * @param fixedColor 比例不对时补充的颜色，不补充为<code>null</code>
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void scale(InputStream srcStream, OutputStream destStream, int width, int height, Color fixedColor) throws IORuntimeException {
//		scale(read(srcStream), getImageOutputStream(destStream), width, height, fixedColor);
//	}
//
//	/**
//	 * 缩放图像（按高度和宽度缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 缩放后的图像目标流
//	 * @param width 缩放后的宽度
//	 * @param height 缩放后的高度
//	 * @param fixedColor 比例不对时补充的颜色，不补充为<code>null</code>
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void scale(ImageInputStream srcStream, ImageOutputStream destStream, int width, int height, Color fixedColor) throws IORuntimeException {
//		scale(read(srcStream), destStream, width, height, fixedColor);
//	}
//
//	/**
//	 * 缩放图像（按高度和宽度缩放）<br>
//	 * 缩放后默认为jpeg格式，此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param destImageStream 缩放后的图像目标流
//	 * @param width 缩放后的宽度
//	 * @param height 缩放后的高度
//	 * @param fixedColor 比例不对时补充的颜色，不补充为<code>null</code>
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void scale(Image srcImage, ImageOutputStream destImageStream, int width, int height, Color fixedColor) throws IORuntimeException {
//		writeJpg(scale(srcImage, width, height, fixedColor), destImageStream);
//	}
//
//	/**
//	 * 缩放图像（按高度和宽度缩放）<br>
//	 * 缩放后默认为jpeg格式
//	 *
//	 * @param srcImage 源图像
//	 * @param width 缩放后的宽度
//	 * @param height 缩放后的高度
//	 * @param fixedColor 比例不对时补充的颜色，不补充为<code>null</code>
//	 * @return {@link Image}
//	 */
//	public static Image scale(Image srcImage, int width, int height, Color fixedColor) {
//		int srcHeight = srcImage.getHeight(null);
//		int srcWidth = srcImage.getWidth(null);
//		double heightRatio = NumberUtil.div(height, srcHeight);
//		double widthRatio = NumberUtil.div(width, srcWidth);
//		if (heightRatio == widthRatio) {
//			// 长宽都按照相同比例缩放时，返回缩放后的图片
//			return scale(srcImage, width, height);
//		}
//
//		Image itemp = null;
//		// 宽缩放比例小就按照宽缩放，否则按照高缩放
//		if (widthRatio < heightRatio) {
//			itemp = scale(srcImage, width, (int) (srcHeight * widthRatio));
//		} else {
//			itemp = scale(srcImage, (int) (srcWidth * heightRatio), height);
//		}
//
//		if (null == fixedColor) {// 补白
//			fixedColor = Color.WHITE;
//		}
//		final BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
//		Graphics2D g = image.createGraphics();
//
//		// 设置背景
//		g.setBackground(fixedColor);
//		g.clearRect(0, 0, width, height);
//
//		final int itempHeight = itemp.getHeight(null);
//		final int itempWidth = itemp.getWidth(null);
//		// 在中间贴图
//		g.drawImage(itemp, (width - itempWidth) / 2, (height - itempHeight) / 2, itempWidth, itempHeight, fixedColor, null);
//
//		g.dispose();
//		itemp = image;
//		return itemp;
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------------- cut
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)
//	 *
//	 * @param srcImgFile 源图像文件
//	 * @param destImgFile 切片后的图像文件
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @since 3.1.0
//	 */
//	public static void cut(File srcImgFile, File destImgFile, Rectangle rectangle) {
//		cut(read(srcImgFile), destImgFile, rectangle);
//	}
//
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)，此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 切片后的图像输出流
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @since 3.1.0
//	 */
//	public static void cut(InputStream srcStream, OutputStream destStream, Rectangle rectangle) {
//		cut(read(srcStream), destStream, rectangle);
//	}
//
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)，此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 切片后的图像输出流
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @since 3.1.0
//	 */
//	public static void cut(ImageInputStream srcStream, ImageOutputStream destStream, Rectangle rectangle) {
//		cut(read(srcStream), destStream, rectangle);
//	}
//
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)，此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param destFile 输出的文件
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @since 3.2.2
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void cut(Image srcImage, File destFile, Rectangle rectangle) throws IORuntimeException {
//		write(cut(srcImage, rectangle), destFile);
//	}
//
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)，此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param out 切片后的图像输出流
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @since 3.1.0
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void cut(Image srcImage, OutputStream out, Rectangle rectangle) throws IORuntimeException {
//		cut(srcImage, getImageOutputStream(out), rectangle);
//	}
//
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)，此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param destImageStream 切片后的图像输出流
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @since 3.1.0
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void cut(Image srcImage, ImageOutputStream destImageStream, Rectangle rectangle) throws IORuntimeException {
//		writeJpg(cut(srcImage, rectangle), destImageStream);
//	}
//
//	/**
//	 * 图像切割(按指定起点坐标和宽高切割)
//	 *
//	 * @param srcImage 源图像
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @return {@link BufferedImage}
//	 * @since 3.1.0
//	 */
//	public static BufferedImage cut(Image srcImage, Rectangle rectangle) {
//		final ImageFilter cropFilter = new CropImageFilter(rectangle.x, rectangle.y, rectangle.width, rectangle.height);
//		Image img = Toolkit.getDefaultToolkit().createImage(new FilteredImageSource(srcImage.getSource(), cropFilter));
//		return toBufferedImage(img);
//	}
//
//	/**
//	 * 图像切片（指定切片的宽度和高度）
//	 *
//	 * @param srcImageFile 源图像
//	 * @param descDir 切片目标文件夹
//	 * @param destWidth 目标切片宽度。默认200
//	 * @param destHeight 目标切片高度。默认150
//	 */
//	public static void slice(File srcImageFile, File descDir, int destWidth, int destHeight) {
//		slice(read(srcImageFile), descDir, destWidth, destHeight);
//	}
//
//	/**
//	 * 图像切片（指定切片的宽度和高度）
//	 *
//	 * @param srcImage 源图像
//	 * @param descDir 切片目标文件夹
//	 * @param destWidth 目标切片宽度。默认200
//	 * @param destHeight 目标切片高度。默认150
//	 */
//	public static void slice(Image srcImage, File descDir, int destWidth, int destHeight) {
//		if (destWidth <= 0) {
//			destWidth = 200; // 切片宽度
//		}
//		if (destHeight <= 0) {
//			destHeight = 150; // 切片高度
//		}
//		int srcWidth = srcImage.getHeight(null); // 源图宽度
//		int srcHeight = srcImage.getWidth(null); // 源图高度
//
//		try {
//			if (srcWidth > destWidth && srcHeight > destHeight) {
//				int cols = 0; // 切片横向数量
//				int rows = 0; // 切片纵向数量
//				// 计算切片的横向和纵向数量
//				if (srcWidth % destWidth == 0) {
//					cols = srcWidth / destWidth;
//				} else {
//					cols = (int) Math.floor(srcWidth / destWidth) + 1;
//				}
//				if (srcHeight % destHeight == 0) {
//					rows = srcHeight / destHeight;
//				} else {
//					rows = (int) Math.floor(srcHeight / destHeight) + 1;
//				}
//				// 循环建立切片
//				BufferedImage tag;
//				for (int i = 0; i < rows; i++) {
//					for (int j = 0; j < cols; j++) {
//						// 四个参数分别为图像起点坐标和宽高
//						// 即: CropImageFilter(int x,int y,int width,int height)
//						tag = cut(srcImage, new Rectangle(j * destWidth, i * destHeight, destWidth, destHeight));
//						// 输出为文件
//						ImageIO.write(tag, IMAGE_TYPE_JPEG, new File(descDir, "_r" + i + "_c" + j + ".jpg"));
//					}
//				}
//			}
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 图像切割（指定切片的行数和列数）
//	 *
//	 * @param srcImageFile 源图像文件
//	 * @param destDir 切片目标文件夹
//	 * @param rows 目标切片行数。默认2，必须是范围 [1, 20] 之内
//	 * @param cols 目标切片列数。默认2，必须是范围 [1, 20] 之内
//	 */
//	public static void sliceByRowsAndCols(File srcImageFile, File destDir, int rows, int cols) {
//		try {
//			sliceByRowsAndCols(ImageIO.read(srcImageFile), destDir, rows, cols);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 图像切割（指定切片的行数和列数）
//	 *
//	 * @param srcImage 源图像
//	 * @param destDir 切片目标文件夹
//	 * @param rows 目标切片行数。默认2，必须是范围 [1, 20] 之内
//	 * @param cols 目标切片列数。默认2，必须是范围 [1, 20] 之内
//	 */
//	public static void sliceByRowsAndCols(Image srcImage, File destDir, int rows, int cols) {
//		if(false == destDir.exists()) {
//			FileUtil.mkdir(destDir);
//		}else if(false == destDir.isDirectory()) {
//			throw new IllegalArgumentException("Destination Dir must be a Directory !");
//		}
//
//		try {
//			if (rows <= 0 || rows > 20) {
//				rows = 2; // 切片行数
//			}
//			if (cols <= 0 || cols > 20) {
//				cols = 2; // 切片列数
//			}
//			// 读取源图像
//			final BufferedImage bi = toBufferedImage(srcImage);
//			int srcWidth = bi.getWidth(); // 源图宽度
//			int srcHeight = bi.getHeight(); // 源图高度
//
//			int destWidth = NumberUtil.partValue(srcWidth, cols); // 每张切片的宽度
//			int destHeight = NumberUtil.partValue(srcHeight, rows); // 每张切片的高度
//
//			// 循环建立切片
//			BufferedImage tag;
//			for (int i = 0; i < rows; i++) {
//				for (int j = 0; j < cols; j++) {
//					tag = cut(bi, new Rectangle(j * destWidth, i * destHeight, destWidth, destHeight));
//					// 输出为文件
//					ImageIO.write(tag, IMAGE_TYPE_JPEG, new File(destDir, "_r" + i + "_c" + j + ".jpg"));
//				}
//			}
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------------- convert
//	/**
//	 * 图像类型转换：GIF=》JPG、GIF=》PNG、PNG=》JPG、PNG=》GIF(X)、BMP=》PNG
//	 *
//	 * @param srcImageFile 源图像文件
//	 * @param formatName 包含格式非正式名称的 String：如JPG、JPEG、GIF等
//	 * @param destImageFile 目标图像文件
//	 */
//	public static void convert(File srcImageFile, String formatName, File destImageFile) {
//		ImageOutputStream imageOutputStream = null;
//		try {
//			imageOutputStream = ImageIO.createImageOutputStream(destImageFile);
//			convert(ImageIO.read(srcImageFile), formatName, imageOutputStream);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		} finally {
//			IoUtil.close(imageOutputStream);
//		}
//	}
//
//	/**
//	 * 图像类型转换：GIF=》JPG、GIF=》PNG、PNG=》JPG、PNG=》GIF(X)、BMP=》PNG<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param formatName 包含格式非正式名称的 String：如JPG、JPEG、GIF等
//	 * @param destStream 目标图像输出流
//	 * @since 3.0.9
//	 */
//	public static void convert(InputStream srcStream, String formatName, OutputStream destStream) {
//		try {
//			convert(ImageIO.read(srcStream), formatName, ImageIO.createImageOutputStream(destStream));
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 图像类型转换：GIF=》JPG、GIF=》PNG、PNG=》JPG、PNG=》GIF(X)、BMP=》PNG<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param formatName 包含格式非正式名称的 String：如JPG、JPEG、GIF等
//	 * @param destStream 目标图像输出流
//	 * @since 3.0.9
//	 */
//	public static void convert(ImageInputStream srcStream, String formatName, ImageOutputStream destStream) {
//		try {
//			convert(ImageIO.read(srcStream), formatName, destStream);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 图像类型转换：GIF=》JPG、GIF=》PNG、PNG=》JPG、PNG=》GIF(X)、BMP=》PNG<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param formatName 包含格式非正式名称的 String：如JPG、JPEG、GIF等
//	 * @param destImageStream 目标图像输出流
//	 * @since 3.0.9
//	 */
//	public static void convert(Image srcImage, String formatName, ImageOutputStream destImageStream) {
//		try {
//			ImageIO.write(toBufferedImage(srcImage), formatName, destImageStream);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------------- grey
//	/**
//	 * 彩色转为黑白
//	 *
//	 * @param srcImageFile 源图像地址
//	 * @param destImageFile 目标图像地址
//	 */
//	public static void gray(File srcImageFile, File destImageFile) {
//		gray(read(srcImageFile), destImageFile);
//	}
//
//	/**
//	 * 彩色转为黑白<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @since 3.0.9
//	 */
//	public static void gray(InputStream srcStream, OutputStream destStream) {
//		gray(read(srcStream), getImageOutputStream(destStream));
//	}
//
//	/**
//	 * 彩色转为黑白<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @since 3.0.9
//	 */
//	public static void gray(ImageInputStream srcStream, ImageOutputStream destStream) {
//		gray(read(srcStream), destStream);
//	}
//
//	/**
//	 * 彩色转为黑白
//	 *
//	 * @param srcImage 源图像流
//	 * @param outFile 目标文件
//	 * @since 3.2.2
//	 */
//	public static void gray(Image srcImage, File outFile) {
//		write(gray(srcImage), outFile);
//	}
//
//	/**
//	 * 彩色转为黑白<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param out 目标图像流
//	 * @since 3.2.2
//	 */
//	public static void gray(Image srcImage, OutputStream out) {
//		gray(srcImage, getImageOutputStream(out));
//	}
//
//	/**
//	 * 彩色转为黑白<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param destImageStream 目标图像流
//	 * @since 3.0.9
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void gray(Image srcImage, ImageOutputStream destImageStream) throws IORuntimeException {
//		writeJpg(gray(srcImage), destImageStream);
//	}
//
//	/**
//	 * 彩色转为黑白
//	 *
//	 * @param srcImage 源图像流
//	 * @return {@link Image}灰度后的图片
//	 * @since 3.1.0
//	 */
//	public static BufferedImage gray(Image srcImage) {
//		BufferedImage grayImage = toBufferedImage(srcImage);
//		final ColorConvertOp op = new ColorConvertOp(ColorSpace.getInstance(ColorSpace.CS_GRAY), null);
//		grayImage = op.filter(grayImage, null);
//		return grayImage;
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------------- binary
//	/**
//	 * 彩色转为黑白二值化图片，根据目标文件扩展名确定转换后的格式
//	 *
//	 * @param srcImageFile 源图像地址
//	 * @param destImageFile 目标图像地址
//	 */
//	public static void binary(File srcImageFile, File destImageFile) {
//		binary(read(srcImageFile), destImageFile);
//	}
//
//	/**
//	 * 彩色转为黑白二值化图片<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @param imageType 图片格式(扩展名)
//	 * @since 4.0.5
//	 */
//	public static void binary(InputStream srcStream, OutputStream destStream, String imageType) {
//		binary(read(srcStream), getImageOutputStream(destStream), imageType);
//	}
//
//	/**
//	 * 彩色转为黑白黑白二值化图片<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @param imageType 图片格式(扩展名)
//	 * @since 4.0.5
//	 */
//	public static void binary(ImageInputStream srcStream, ImageOutputStream destStream, String imageType) {
//		binary(read(srcStream), destStream, imageType);
//	}
//
//	/**
//	 * 彩色转为黑白二值化图片，根据目标文件扩展名确定转换后的格式
//	 *
//	 * @param srcImage 源图像流
//	 * @param outFile 目标文件
//	 * @since 4.0.5
//	 */
//	public static void binary(Image srcImage, File outFile) {
//		write(binary(srcImage), outFile);
//	}
//
//	/**
//	 * 彩色转为黑白二值化图片<br>
//	 * 此方法并不关闭流，输出JPG格式
//	 * @param srcImage 源图像流
//	 * @param out 目标图像流
//	 * @param imageType 图片格式(扩展名)
//	 * @since 4.0.5
//	 */
//	public static void binary(Image srcImage, OutputStream out, String imageType) {
//		binary(srcImage, getImageOutputStream(out), imageType);
//	}
//
//	/**
//	 * 彩色转为黑白二值化图片<br>
//	 * 此方法并不关闭流，输出JPG格式
//	 *
//	 * @param srcImage 源图像流
//	 * @param destImageStream 目标图像流
//	 * @param imageType 图片格式(扩展名)
//	 * @since 4.0.5
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void binary(Image srcImage, ImageOutputStream destImageStream, String imageType) throws IORuntimeException {
//		write(binary(srcImage), imageType, destImageStream);
//	}
//
//	/**
//	 * 彩色转为黑白二值化图片
//	 *
//	 * @param srcImage 源图像流
//	 * @return {@link Image}二值化后的图片
//	 * @since 4.0.5
//	 */
//	public static BufferedImage binary(Image srcImage) {
//		return copyImage(srcImage, BufferedImage.TYPE_BYTE_BINARY);
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------------- press
//	/**
//	 * 给图片添加文字水印
//	 *
//	 * @param imageFile 源图像文件
//	 * @param destFile 目标图像文件
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 */
//	public static void pressText(File imageFile, File destFile, String pressText, Color color, Font font, int x, int y, float alpha) {
//		pressText(read(imageFile), destFile, pressText, color, font, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加文字水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 */
//	public static void pressText(InputStream srcStream, OutputStream destStream, String pressText, Color color, Font font, int x, int y, float alpha) {
//		pressText(read(srcStream), getImageOutputStream(destStream), pressText, color, font, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加文字水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 */
//	public static void pressText(ImageInputStream srcStream, ImageOutputStream destStream, String pressText, Color color, Font font, int x, int y, float alpha) {
//		pressText(read(srcStream), destStream, pressText, color, font, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加文字水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param destFile 目标流
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void pressText(Image srcImage, File destFile, String pressText, Color color, Font font, int x, int y, float alpha) throws IORuntimeException {
//		write(pressText(srcImage, pressText, color, font, x, y, alpha), destFile);
//	}
//
//	/**
//	 * 给图片添加文字水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param to 目标流
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void pressText(Image srcImage, OutputStream to, String pressText, Color color, Font font, int x, int y, float alpha) throws IORuntimeException {
//		pressText(srcImage, getImageOutputStream(to), pressText, color, font, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加文字水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param destImageStream 目标图像流
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void pressText(Image srcImage, ImageOutputStream destImageStream, String pressText, Color color, Font font, int x, int y, float alpha) throws IORuntimeException {
//		writeJpg(pressText(srcImage, pressText, color, font, x, y, alpha), destImageStream);
//	}
//
//	/**
//	 * 给图片添加文字水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像
//	 * @param pressText 水印文字
//	 * @param color 水印的字体颜色
//	 * @param font {@link Font} 字体相关信息，如果默认则为{@code null}
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @return 处理后的图像
//	 * @since 3.2.2
//	 */
//	public static BufferedImage pressText(Image srcImage, String pressText, Color color, Font font, int x, int y, float alpha) {
//		int width = srcImage.getWidth(null);
//		int height = srcImage.getHeight(null);
//		BufferedImage destImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
//		Graphics2D g = destImage.createGraphics();
//
//		//抗锯齿
//		g.setRenderingHint(RenderingHints.KEY_ANTIALIASING,RenderingHints.VALUE_ANTIALIAS_ON);
//		// 绘制背景
//		g.drawImage(srcImage, 0, 0, width, height, null);
//		g.setColor(color);
//		g.setFont(font);
//		g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, alpha));
//		// 在指定坐标绘制水印文字
//		final int fontSize = font.getSize();
//		g.drawString(pressText, (width - (getLength(pressText) * fontSize)) / 2 + x, (height - fontSize) / 2 + y);
//		g.dispose();
//
//		return destImage;
//	}
//
//	/**
//	 * 给图片添加图片水印
//	 *
//	 * @param srcImageFile 源图像文件
//	 * @param destImageFile 目标图像文件
//	 * @param pressImg 水印图片
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 */
//	public static void pressImage(File srcImageFile, File destImageFile, Image pressImg, int x, int y, float alpha) {
//		pressImage(read(srcImageFile), destImageFile, pressImg, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加图片水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @param pressImg 水印图片，可以使用{@link ImageIO#read(File)}方法读取文件
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 */
//	public static void pressImage(InputStream srcStream, OutputStream destStream, Image pressImg, int x, int y, float alpha) {
//		pressImage(read(srcStream), getImageOutputStream(destStream), pressImg, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加图片水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcStream 源图像流
//	 * @param destStream 目标图像流
//	 * @param pressImg 水印图片，可以使用{@link ImageIO#read(File)}方法读取文件
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void pressImage(ImageInputStream srcStream, ImageOutputStream destStream, Image pressImg, int x, int y, float alpha) throws IORuntimeException {
//		pressImage(read(srcStream), destStream, pressImg, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加图片水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param outFile 写出文件
//	 * @param pressImg 水印图片，可以使用{@link ImageIO#read(File)}方法读取文件
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void pressImage(Image srcImage, File outFile, Image pressImg, int x, int y, float alpha) throws IORuntimeException {
//		write(pressImage(srcImage, pressImg, x, y, alpha), outFile);
//	}
//
//	/**
//	 * 给图片添加图片水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param out 目标图像流
//	 * @param pressImg 水印图片，可以使用{@link ImageIO#read(File)}方法读取文件
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void pressImage(Image srcImage, OutputStream out, Image pressImg, int x, int y, float alpha) throws IORuntimeException {
//		pressImage(srcImage, getImageOutputStream(out), pressImg, x, y, alpha);
//	}
//
//	/**
//	 * 给图片添加图片水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param destImageStream 目标图像流
//	 * @param pressImg 水印图片，可以使用{@link ImageIO#read(File)}方法读取文件
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void pressImage(Image srcImage, ImageOutputStream destImageStream, Image pressImg, int x, int y, float alpha) throws IORuntimeException {
//		writeJpg(pressImage(srcImage, pressImg, x, y, alpha), destImageStream);
//	}
//
//	/**
//	 * 给图片添加图片水印<br>
//	 * 此方法并不关闭流
//	 *
//	 * @param srcImage 源图像流
//	 * @param pressImg 水印图片，可以使用{@link ImageIO#read(File)}方法读取文件
//	 * @param x 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param y 修正值。 默认在中间，偏移量相对于中间偏移
//	 * @param alpha 透明度：alpha 必须是范围 [0.0, 1.0] 之内（包含边界值）的一个浮点数字
//	 * @return 结果图片
//	 */
//	public static BufferedImage pressImage(Image srcImage, Image pressImg, int x, int y, float alpha) {
//		final BufferedImage destImg = toBufferedImage(srcImage);
//		final int width = destImg.getWidth();
//		final int height = destImg.getHeight();
//
//		int pressImgWidth = pressImg.getWidth(null);
//		int pressImgHeight = pressImg.getHeight(null);
//		x += (width - pressImgWidth) / 2;
//		y += (height - pressImgHeight) / 2;
//
//		draw(destImg, pressImg, new Rectangle(x, y, pressImgWidth, pressImgHeight), alpha);
//		return destImg;
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------------- rotate
//	/**
//	 * 旋转图片为指定角度<br>
//	 * 此方法不会关闭输出流
//	 *
//	 * @param imageFile 被旋转图像文件
//	 * @param degree 旋转角度
//	 * @param outFile 输出文件
//	 * @since 3.2.2
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void rotate(File imageFile, int degree, File outFile) throws IORuntimeException {
//		rotate(read(imageFile), degree, outFile);
//	}
//
//	/**
//	 * 旋转图片为指定角度<br>
//	 * 此方法不会关闭输出流
//	 *
//	 * @param image 目标图像
//	 * @param degree 旋转角度
//	 * @param outFile 输出文件
//	 * @since 3.2.2
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void rotate(Image image, int degree, File outFile) throws IORuntimeException {
//		write(rotate(image, degree), outFile);
//	}
//
//	/**
//	 * 旋转图片为指定角度<br>
//	 * 此方法不会关闭输出流
//	 *
//	 * @param image 目标图像
//	 * @param degree 旋转角度
//	 * @param out 输出流
//	 * @since 3.2.2
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void rotate(Image image, int degree, OutputStream out) throws IORuntimeException {
//		writeJpg(rotate(image, degree), getImageOutputStream(out));
//	}
//
//	/**
//	 * 旋转图片为指定角度<br>
//	 * 此方法不会关闭输出流，输出格式为JPG
//	 *
//	 * @param image 目标图像
//	 * @param degree 旋转角度
//	 * @param out 输出图像流
//	 * @since 3.2.2
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void rotate(Image image, int degree, ImageOutputStream out) throws IORuntimeException {
//		writeJpg(rotate(image, degree), out);
//	}
//
//	/**
//	 * 旋转图片为指定角度<br>
//	 * 来自：http://blog.51cto.com/cping1982/130066
//	 *
//	 * @param image 目标图像
//	 * @param degree 旋转角度
//	 * @return 旋转后的图片
//	 * @since 3.2.2
//	 */
//	public static BufferedImage rotate(Image image, int degree) {
//		int width = image.getWidth(null);
//		int height = image.getHeight(null);
//		int type = toBufferedImage(image).getTransparency();
//
//		final BufferedImage destImg = new BufferedImage(width, height, type);
//		Graphics2D graphics2d = destImg.createGraphics();
//		// 抗锯齿
//		graphics2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);
//		// 从中心旋转
//		graphics2d.rotate(Math.toRadians(degree), width / 2, height / 2);
//		graphics2d.drawImage(image, 0, 0, null);
//		graphics2d.dispose();
//		return destImg;
//	}

//	// ---------------------------------------------------------------------------------------------------------------------- flip
//	/**
//	 * 水平翻转图像
//	 *
//	 * @param imageFile 图像文件
//	 * @param outFile 输出文件
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static void flip(File imageFile, File outFile) throws IORuntimeException {
//		flip(read(imageFile), outFile);
//	}

    /**
     * 水平翻转图像
     *
     * @param image   图像
     * @param outFile 输出文件
     * @throws IORuntimeException IO异常
     * @since 3.2.2
     */
    public static void flip(Image image, File outFile) throws IORuntimeException {
        write(flip(image), outFile);
    }

    /**
     * 水平翻转图像
     *
     * @param image 图像
     * @param out   输出
     * @throws IORuntimeException IO异常
     * @since 3.2.2
     */
    public static void flip(Image image, OutputStream out) throws IORuntimeException {
        flip(image, getImageOutputStream(out));
    }

    /**
     * 水平翻转图像，写出格式为JPG
     *
     * @param image 图像
     * @param out   输出
     * @throws IORuntimeException IO异常
     * @since 3.2.2
     */
    public static void flip(Image image, ImageOutputStream out) throws IORuntimeException {
        writeJpg(flip(image), out);
    }

    /**
     * 水平翻转图像
     *
     * @param image 图像
     * @return 翻转后的图片
     * @since 3.2.2
     */
    public static BufferedImage flip(Image image) {
        int width = image.getWidth(null);
        int height = image.getHeight(null);

        BufferedImage img = new BufferedImage(width, height, toBufferedImage(image).getTransparency());
        Graphics2D graphics2d;
        (graphics2d = img.createGraphics()).drawImage(image, 0, 0, width, height, width, 0, 0, height, null);
        graphics2d.dispose();
        return img;
    }

    // ---------------------------------------------------------------------------------------------------------------------- other

    /**
     * {@link Image} 转 {@link BufferedImage}<br>
     * 首先尝试强转，否则新建一个{@link BufferedImage}后重新绘制
     *
     * @param img {@link Image}
     * @return {@link BufferedImage}
     */
    public static BufferedImage toBufferedImage(Image img) {
        if (img instanceof BufferedImage) {
            return (BufferedImage) img;
        }

        return copyImage(img, BufferedImage.TYPE_INT_RGB);
    }

    /**
     * 将已有Image复制新的一份出来
     *
     * @param img       {@link Image}
     * @param imageType {@link BufferedImage}中的常量，图像类型，例如黑白等
     * @return {@link BufferedImage}
     * @see BufferedImage#TYPE_INT_RGB
     * @see BufferedImage#TYPE_INT_ARGB
     * @see BufferedImage#TYPE_INT_ARGB_PRE
     * @see BufferedImage#TYPE_INT_BGR
     * @see BufferedImage#TYPE_3BYTE_BGR
     * @see BufferedImage#TYPE_4BYTE_ABGR
     * @see BufferedImage#TYPE_4BYTE_ABGR_PRE
     * @see BufferedImage#TYPE_BYTE_GRAY
     * @see BufferedImage#TYPE_USHORT_GRAY
     * @see BufferedImage#TYPE_BYTE_BINARY
     * @see BufferedImage#TYPE_BYTE_INDEXED
     * @see BufferedImage#TYPE_USHORT_565_RGB
     * @see BufferedImage#TYPE_USHORT_555_RGB
     */
    public static BufferedImage copyImage(Image img, int imageType) {
        final BufferedImage bimage = new BufferedImage(img.getWidth(null), img.getHeight(null), imageType);
        final Graphics2D bGr = bimage.createGraphics();
        bGr.drawImage(img, 0, 0, null);
        bGr.dispose();

        return bimage;
    }

//	/**
//	 * 将Base64编码的图像信息转为 {@link BufferedImage}
//	 *
//	 * @param base64 图像的Base64表示
//	 * @return {@link BufferedImage}
//	 * @throws IORuntimeException IO异常
//	 */
//	public static BufferedImage toImage(String base64) throws IORuntimeException {
//		byte[] decode = Base64.decode(base64, CharsetUtil.CHARSET_UTF_8);
//		return toImage(decode);
//	}

//	/**
//	 * 将的图像bytes转为 {@link BufferedImage}
//	 *
//	 * @param imageBytes 图像bytes
//	 * @return {@link BufferedImage}
//	 * @throws IORuntimeException IO异常
//	 */
//	public static BufferedImage toImage(byte[] imageBytes) throws IORuntimeException {
//		return read(new ByteArrayInputStream(imageBytes));
//	}
//
//	/**
//	 * 根据文字创建PNG图片
//	 *
//	 * @param str 文字
//	 * @param font 字体{@link Font}
//	 * @param backgroundColor 背景颜色
//	 * @param fontColor 字体颜色
//	 * @param out 图片输出地
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void createImage(String str, Font font, Color backgroundColor, Color fontColor, ImageOutputStream out) throws IORuntimeException {
//		// 获取font的样式应用在str上的整个矩形
//		Rectangle2D r = font.getStringBounds(str, new FontRenderContext(AffineTransform.getScaleInstance(1, 1), false, false));
//		int unitHeight = (int) Math.floor(r.getHeight());// 获取单个字符的高度
//		// 获取整个str用了font样式的宽度这里用四舍五入后+1保证宽度绝对能容纳这个字符串作为图片的宽度
//		int width = (int) Math.round(r.getWidth()) + 1;
//		int height = unitHeight + 3;// 把单个字符的高度+3保证高度绝对能容纳字符串作为图片的高度
//		// 创建图片
//		BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_BGR);
//		Graphics g = image.getGraphics();
//		g.setColor(backgroundColor);
//		g.fillRect(0, 0, width, height);// 先用背景色填充整张图片,也就是背景
//		g.setColor(fontColor);
//		g.setFont(font);// 设置画笔字体
//		g.drawString(str, 0, font.getSize());// 画出字符串
//		g.dispose();
//		writePng(image, out);
//	}

    /**
     * 根据文件创建字体<br>
     * 首先尝试创建{@link Font#TRUETYPE_FONT}字体，此类字体无效则创建{@link Font#TYPE1_FONT}
     *
     * @param fontFile 字体文件
     * @return {@link Font}
     * @since 3.0.9
     */
//	public static Font createFont(File fontFile) {
//		try {
//			return Font.createFont(Font.TRUETYPE_FONT, fontFile);
//		} catch (FontFormatException e) {
//			// True Type字体无效时使用Type1字体
//			try {
//				return Font.createFont(Font.TYPE1_FONT, fontFile);
//			} catch (Exception e1) {
//				throw new UtilException(e);
//			}
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}

//	/**
//	 * 根据文件创建字体<br>
//	 * 首先尝试创建{@link Font#TRUETYPE_FONT}字体，此类字体无效则创建{@link Font#TYPE1_FONT}
//	 *
//	 * @param fontStream 字体流
//	 * @return {@link Font}
//	 * @since 3.0.9
//	 */
//	public static Font createFont(InputStream fontStream) {
//		try {
//			return Font.createFont(Font.TRUETYPE_FONT, fontStream);
//		} catch (FontFormatException e) {
//			// True Type字体无效时使用Type1字体
//			try {
//				return Font.createFont(Font.TYPE1_FONT, fontStream);
//			} catch (Exception e1) {
//				throw new UtilException(e1);
//			}
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}

    /**
     * 创建{@link Graphics2D}
     *
     * @param image {@link BufferedImage}
     * @param color {@link Color}背景颜色以及当前画笔颜色
     * @return {@link Graphics2D}
     * @since 3.2.3
     */
    public static Graphics2D createGraphics(BufferedImage image, Color color) {
        final Graphics2D g = image.createGraphics();
        // 填充背景
        g.setColor(color);
        g.fillRect(0, 0, image.getWidth(), image.getHeight());

        return g;
    }

    /**
     * 写出图像为JPG格式
     *
     * @param image           {@link Image}
     * @param destImageStream 写出到的目标流
     * @throws IORuntimeException IO异常
     */
    public static void writeJpg(Image image, ImageOutputStream destImageStream) throws IORuntimeException {
        write(image, IMAGE_TYPE_JPG, destImageStream);
    }
//
//	/**
//	 * 写出图像为PNG格式
//	 *
//	 * @param image {@link Image}
//	 * @param destImageStream 写出到的目标流
//	 * @throws IORuntimeException IO异常
//	 */
//	public static void writePng(Image image, ImageOutputStream destImageStream) throws IORuntimeException {
//		write(image, IMAGE_TYPE_PNG, destImageStream);
//	}
//
//	/**
//	 * 写出图像为JPG格式
//	 *
//	 * @param image {@link Image}
//	 * @param out 写出到的目标流
//	 * @throws IORuntimeException IO异常
//	 * @since 4.0.10
//	 */
//	public static void writeJpg(Image image, OutputStream out) throws IORuntimeException {
//		write(image, IMAGE_TYPE_JPG, out);
//	}
//
//	/**
//	 * 写出图像为PNG格式
//	 *
//	 * @param image {@link Image}
//	 * @param out 写出到的目标流
//	 * @throws IORuntimeException IO异常
//	 * @since 4.0.10
//	 */
//	public static void writePng(Image image, OutputStream out) {
//		write(image, IMAGE_TYPE_PNG, out);
//	}

    /**
     * 写出图像为PNG格式
     *
     * @param image     {@link Image}
     * @param imageType 图片类型（图片扩展名）
     * @param out       写出到的目标流
     * @throws IORuntimeException IO异常
     * @since 3.1.2
     */
    public static void write(Image image, String imageType, OutputStream out) {
        write(image, imageType, getImageOutputStream(out));
    }

    /**
     * 写出图像为PNG格式
     *
     * @param image           {@link Image}
     * @param imageType       图片类型（图片扩展名）
     * @param destImageStream 写出到的目标流
     * @throws IORuntimeException IO异常
     * @since 3.1.2
     */
    public static void write(Image image, String imageType, ImageOutputStream destImageStream) throws IORuntimeException {
        try {
            ImageIO.write(toBufferedImage(image), imageType, destImageStream);// 输出到文件流
        } catch (IOException e) {
            throw new IORuntimeException(e);
        }
    }

    /**
     * 写出图像为目标文件扩展名对应的格式
     *
     * @param image      {@link Image}
     * @param targetFile 目标文件
     * @throws IORuntimeException IO异常
     * @since 3.1.0
     */
    public static void write(Image image, File targetFile) throws IORuntimeException {
        String formatName = FileUtil.extName(targetFile);
        if (StrUtil.isBlank(formatName)) {
            formatName = IMAGE_TYPE_JPG;
        }
        try {
            ImageIO.write(toBufferedImage(image), formatName, targetFile);// 输出到文件
        } catch (IOException e) {
            throw new IORuntimeException(e);
        }
    }

    /**
     * 获得{@link ImageReader}
     *
     * @param type 图片文件类型，例如 "jpeg" 或 "tiff"
     * @return {@link ImageReader}
     */
    public static ImageReader getReader(String type) {
        final Iterator<ImageReader> iterator = ImageIO.getImageReadersByFormatName(type);
        if (iterator.hasNext()) {
            return iterator.next();
        }
        return null;
    }
//
//	/**
//	 * 从文件中读取图片
//	 *
//	 * @param imageFile 图片文件
//	 * @return 图片
//	 * @since 3.2.2
//	 */
//	public static BufferedImage read(File imageFile) {
//		try {
//			return ImageIO.read(imageFile);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 从流中读取图片
//	 *
//	 * @param imageStream 图片文件
//	 * @return 图片
//	 * @since 3.2.2
//	 */
//	public static BufferedImage read(InputStream imageStream) {
//		try {
//			return ImageIO.read(imageStream);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 从图片流中读取图片
//	 *
//	 * @param imageStream 图片文件
//	 * @return 图片
//	 * @since 3.2.2
//	 */
//	public static BufferedImage read(ImageInputStream imageStream) {
//		try {
//			return ImageIO.read(imageStream);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 从URL中读取图片
//	 *
//	 * @param imageUrl 图片文件
//	 * @return 图片
//	 * @since 3.2.2
//	 */
//	public static BufferedImage read(URL imageUrl) {
//		try {
//			return ImageIO.read(imageUrl);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//

    /**
     * 获取{@link ImageOutputStream}
     *
     * @param out {@link OutputStream}
     * @return {@link ImageOutputStream}
     * @throws IORuntimeException IO异常
     * @since 3.1.2
     */
    public static ImageOutputStream getImageOutputStream(OutputStream out) throws IORuntimeException {
        try {
            return ImageIO.createImageOutputStream(out);
        } catch (IOException e) {
            throw new IORuntimeException(e);
        }
    }
//
//	/**
//	 * 获取{@link ImageOutputStream}
//	 *
//	 * @param outFile {@link File}
//	 * @return {@link ImageOutputStream}
//	 * @throws IORuntimeException IO异常
//	 * @since 3.2.2
//	 */
//	public static ImageOutputStream getImageOutputStream(File outFile) throws IORuntimeException {
//		try {
//			return ImageIO.createImageOutputStream(outFile);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 获取{@link ImageInputStream}
//	 *
//	 * @param in {@link InputStream}
//	 * @return {@link ImageInputStream}
//	 * @throws IORuntimeException IO异常
//	 * @since 3.1.2
//	 */
//	public static ImageInputStream getImageInputStream(InputStream in) throws IORuntimeException {
//		try {
//			return ImageIO.createImageInputStream(in);
//		} catch (IOException e) {
//			throw new IORuntimeException(e);
//		}
//	}
//
//	/**
//	 * 生成随机颜色
//	 *
//	 * @return 随机颜色
//	 * @since 3.1.2
//	 */
//	public static Color randomColor() {
//		return randomColor(null);
//	}
//
//	/**
//	 * 生成随机颜色
//	 *
//	 * @param random 随机对象 {@link Random}
//	 * @return 随机颜色
//	 * @since 3.1.2
//	 */
//	public static Color randomColor(Random random) {
//		if (null == random) {
//			random = RandomUtil.getRandom();
//		}
//		return new Color(random.nextInt(255), random.nextInt(255), random.nextInt(255));
//	}
//
//	// ---------------------------------------------------------------------------------------------------------------- Private method start
//	/**
//	 * 将图片绘制在背景上
//	 *
//	 * @param backgroundImg 背景图片
//	 * @param img 要绘制的图片
//	 * @param rectangle 矩形对象，表示矩形区域的x，y，width，height
//	 * @return 绘制后的背景
//	 */
//	private static BufferedImage draw(BufferedImage backgroundImg, Image img, Rectangle rectangle, float alpha) {
//		final Graphics2D g = backgroundImg.createGraphics();
//		g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, alpha));
//		g.drawImage(img, rectangle.x, rectangle.y, rectangle.width, rectangle.height, null); // 绘制切割后的图
//		g.dispose();
//		return backgroundImg;
//	}
//
//	/**
//	 * 计算text的长度（一个中文算两个字符）<br>
//	 * 如：text="中国",返回 2；text="test",返回 2；text="中国ABC",返回 4.
//	 *
//	 * @param text 文本
//	 * @return 字符长度
//	 */
//	private static int getLength(String text) {
//		int length = 0;
//		for (int i = 0; i < text.length(); i++) {
//			if (String.valueOf(text.charAt(i)).getBytes().length > 1) {
//				length += 2;
//			} else {
//				length += 1;
//			}
//		}
//		return length / 2;
//	}
    // ---------------------------------------------------------------------------------------------------------------- Private method end
}
